import { login, logout, getInfo } from '@/api/user'
import { getToken, setToken, removeToken } from '@/utils/auth'
import { resetRouter, asyncRoutes, constantRoutes } from '@/router'
import cloneDeep from 'lodash/cloneDeep'
import router from '@/router';
const getDefaultState = () => {
  return {
    // 获取token
    token: getToken(),
    // 存储用户名
    name: '',
    // 用户头像
    avatar: '',
    // 用户菜单数据
    routes: [],
    // 角色信息
    roles: [],
    //对比之后【项目中已有的异步路由，与服务器返回的标记信息进行对比最终需要展示的路由】
    resultAsyncRoutes: [],
    //用户最终需要展示全部路由
    resultAllRputes: []
  }
}

const state = getDefaultState()

const mutations = {
  RESET_STATE: (state) => {
    Object.assign(state, getDefaultState())
  },
  SET_TOKEN: (state, token) => {
    state.token = token
  },

  // 存储用户信息
  SET_USERINFO(state, userInfo) {
    state.name = userInfo.name
    state.avatar = userInfo.avatar
    //菜单权限标记
    state.routes = userInfo.routes;
    //按钮权限标记
    state.buttons = userInfo.buttons;
    //角色
    state.roles = userInfo.roles;
  },
  // 设置要显示的路由
  SET_ROUTERS(state, asyncRoutes) {
    console.log(asyncRoutes);
    // /vuex保存当前用户的异步路由，注意，一个用户需要展示完成路由：常量、异步、任意路由
    state.resultAsyncRoutes = asyncRoutes
    state.resultAllRputes = constantRoutes.concat(state.resultAsyncRoutes);
    router.addRoutes(state.resultAllRputes)
  }
}

// 定义一个函数来循环获取需要显示的路由 当前写的路由和服务器传的路由
function computedAsyncRoutes(asyncRoutes, routers) {
  // console.log(asyncRoutes, routers);
  //过滤出当前用户【超级管理|普通员工】需要展示的异步路由
  return asyncRoutes.filter(item => {
    //数组当中没有这个元素返回索引值-1，如果有这个元素返回的索引值一定不是-1 
    if (routers.indexOf(item.name) != -1) {
      //递归:别忘记还有2、3、4、5、6级路由
      if (item.children && item.children.length) {
        item.children = computedAsyncRoutes(item.children, routers)
      }
      return true;
    }
  })
}


const actions = {
  // 登录
  async login({ commit }, userInfo) {
    // 解构传来的账号密码
    const { username, password } = userInfo
    let res = await login({ username: username.trim(), password: password })
    console.log(res);
    if (res.code == 20000) {
      commit('SET_TOKEN', res.data.token)
      setToken(res.data.token)
      return 'ok'
    } else {
      return Promise.reject('失败')
    }


  },

  // get user info 获取用户信息
  getInfo({ commit, state }) {
    return new Promise((resolve, reject) => {
      getInfo(state.token).then(response => {
        // 获取了用户的所有信息 包括 token 用户名 头像 路由
        const { data } = response
        // console.log(data);
        if (!data) {
          return reject('Verification failed, please Login again.')
        }

        commit('SET_USERINFO', data)
        // 添加路由
        commit('SET_ROUTERS', computedAsyncRoutes(cloneDeep(asyncRoutes), data.routes));
        resolve(data)
      }).catch(error => {
        reject(error)
      })
    })
  },

  // user logout
  logout({ commit, state }) {
    return new Promise((resolve, reject) => {
      logout(state.token).then(() => {
        removeToken() // must remove  token  first
        resetRouter()
        commit('RESET_STATE')
        resolve()
      }).catch(error => {
        reject(error)
      })
    })
  },

  // remove token
  resetToken({ commit }) {
    return new Promise(resolve => {
      removeToken() // must remove  token  first
      commit('RESET_STATE')
      resolve()
    })
  }
}

export default {
  namespaced: true,
  state,
  mutations,
  actions
}

